查看原文
其他

妈蛋!空降的老大叫咱们写得代码全部都得要Mock,看来得996了~

D哥 Java面试那些事儿 2021-09-05

最近,D哥这边新来的技术老大,要求写得所有的代码必须Mock,没有达到90%,都得要打回来~ 


说实话,咱们组之前的任务比较重,能写完代码,在加点测试用例,就很nice了,这么一搞,看来又得要开始加班搞了,谁叫咱们是苦逼的码畜,男人当牛用,女人当男人用,说到底,大多数人跟D哥一样,因为穷才做程序员。。。


说实话,D哥去查看了一下市面上的mock工具,说实话,挺多的,说白了,按照实现原理,主要分为3类:


  • 动态代理:Mockito、EasyMock、MockRunner

  • 自定义类加载器:PowerMock

  • 运行时字节码修改:JMockit、TestableMock


如下图:



动态代理:这种实现方式()只能在类的外围修修补补,不能改动类本身,因此,最安全,但功能也最弱。关于动态代理面试必问,建议多去研究一下。


自定义类加载和字节码修改:这两种方式都会修改类的字节码,前者通过完全接管类的加载过程中来实现,后者则是类加载完成后再对字节码进行“二次改造”;两者在功能上而言,差别并不大(见上图)。

因此,综合来看,阿里的 TestableMock 看起来似乎更牛逼!


使用起来真的方便,正如它官网说得“换种思路写Mock,让单元测试更简单”。


无需初始化,不挑服务框架,甭管要换的是私有方法、静态方法、构造方法还是其他任何类的任何方法,也甭管要换的对象是怎么创建的。写好Mock定义,加个@MockMethod注解,一切统统搞定。


官网源码:https://github.com/alibaba/testable-mock


官方文档:https://alibaba.github.io/testable-mock


# 实现原理


它是基于运行时修改字节码,再通过扫描测试类中是否有@MockMethod、@MockConstructor等注解修饰的方法,来判断是否要进行对应的初始化,与测试框架实现完全解藕,体验上毫无入侵感。


# 如何上手?


1、引入相关依赖

<properties> <java.version>1.8</java.version> <testable.version>0.5.2</testable.version></properties>
<dependencies> <dependency> <groupId>com.alibaba.testable</groupId> <artifactId>testable-all</artifactId> <version>${testable.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>org.junit.jupiter</groupId> <artifactId>junit-jupiter-api</artifactId> <version>${junit.version}</version> <scope>test</scope> </dependency></dependencies><build> <plugins> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-surefire-plugin</artifactId> <version>3.0.0-M5</version> <configuration> <argLine>@{argLine} -javaagent:${settings.localRepository}/com/alibaba/testable/testable-agent/${testable.version}/testable-agent-${testable.version}.jar</argLine> </configuration> </plugin> </plugins></build>


2、增加一个类,调用任意方法、成员方法、静态方法,结合官方demo改动了一下,大家将就看吧,实例如下:


DemoMock.java

/** * 演示基本的Mock功能 * @from 公众号:Java面试那些事儿 * @author D哥 */public class DemoMock {
/** * 普通方式 * @return */ public String commonFunc() { return "idea ".trim() + "." + " studycoder.com".substring(1) + "_" + "idea.studycoder.com".startsWith("ab"); }

public String outerFunc(String website) { return innerFunc(staticFunc()); }

/** * 静态方法 * @return */ private static String staticFunc() { return "idea_studycoder_com".replace("_","."); }
/** * 内部方法 * @param website * @return */ private String innerFunc(String website) { return "our website is: " + website; }}


DemoMockTest.java


public class DemoMockTest {
private DemoMock demoMock = new DemoMock();
public static class Mock {
@MockMethod(targetClass = DemoMock.class) private String innerFunc(String text) { return "mock_" + text; }
@MockMethod(targetClass = DemoMock.class) private String staticFunc() { return "static"; }
@MockMethod(targetClass = String.class) private String trim() { return "idea"; }
@MockMethod(targetClass = String.class, targetMethod = "substring") private String sub(int i) { return "sub"; }
@MockMethod(targetClass = String.class) private boolean startsWith(String s) { return false; } }
@Test void should_able_to_mock_common_method() { assertEquals("idea.sub_false", demoMock.commonFunc()); verify("trim").withTimes(1); verify("sub").withTimes(1); verify("startsWith").withTimes(1); }
@Test void should_able_to_mock_member_method() throws Exception { assertEquals("mock_static", demoMock.outerFunc("hello")); verify("innerFunc").with("static"); verify("staticFunc").with(); }}


# 总结


从上面的实例中,咱们可以看出,相比其它的mock工具,阿里的TestableMock 写单元测试会方便很多,代码量也少了不少,但是也有一些缺点,比如一个方法多次调用返回不同值,需借助TestableTool.MOCK_CONTEXT,对代码来说有一定的侵入性,但总得来说,无不是一种最优的选择。


啥也不说了,D哥决定建议团队的兄弟们用起来了~


你们经常用哪款 mock 工具呢?欢迎在留言区告诉D哥



热门推荐:


: . Video Mini Program Like ,轻点两下取消赞 Wow ,轻点两下取消在看

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存